import axios from 'axios'
import store from '@/store'
import { Message } from 'element-ui'
import router from '@/router'
import qs from 'qs'
const request = axios.create({
// 配置选项
// baseURL
// timeout
})
// 请求拦截器
request.interceptors.request.use(function (config) {
  // 在发送请求之前做些什么
  console.log(config, '接口进来了')
  // 我们就在这里通过改写config配置信息来实现业务功能的统一处理
  const { user } = store.state
  if (user && user.access_token) {
    config.headers.Authorization = user.access_token
  }
  // 注意：这里一定要返回config，否则请求就发不出去了
  return config
}, function (error) {
  // 对请求错误做些什么
  return Promise.reject(error)
})
// 响应拦截器
let isRefreshing = false // 控制刷新 token 的状态
let requests = [] // 存储刷新 token 期间过来的 401 请求
request.interceptors.response.use(function (response) {
  // 状态码为2xx都会进这里
  // 如果是自定义错误状态码，错误处理就写到这里
  return response
}, async function (error) {
  // 超出2xx状态码都执行这里
  // console.log('请求响应失败了 ==> ', error)
  // 如果是使用的 HTTP 状态码，错误处理就写到这里
  // console.dir(error)
  if (error.response) {
    // 请求收到响应了，但是状态码超出了2xx范围
    // 400 401 403 404 405 500
    const { status } = error.response
    if (status === 400) {
      Message.error('请求参数错误')
    } else if (status === 401) {
      // token无效（没有提供token、token是无效的、token过期了）
      // 如果有refresh_token，则尝试使用refresh_token 获取新的 access_token
      //    成功了 -> 把本次失败的请求重新发出去
      //    失败了 -> 跳转登录页重新登录获取新的 token
      // 如果没有，则直接跳转登录页
      if (!store.state.user) {
        redirectLogin()
        return Promise.reject(error)
      }
      // 尝试刷新获取新的token
      if (!isRefreshing) {
        isRefreshing = true // 开启刷新状态
        return refreshToken()
          .then(res => {
            if (!res.data.success) {
              throw new Error('刷新 Token 失败')
            }
            // 无痛刷新，用户无感知（刷新token只能使用一次）
            // 把刷新拿到的新的access_token更新到容器和本地存储中
            // 刷新 token 成功了
            store.commit('setUser', res.data.content)
            // 把 requests 队列中的请求重新发出去
            requests.forEach(cb => cb())
            // 重置requests 数据
            requests = []
            //    成功了 -> 把本次失败的请求重新发出去
            // console.log(error.config) // 失败请求的配置信息
            return request(error.config)
          })
          .catch(err => {
            console.log(err)
            // 把当前登录用户状态清除
            store.commit('setUser', null)
            //    失败了 -> 跳转登录页重新登录获取新的 token
            redirectLogin()
            return Promise.reject(error)
          })
          .finally(() => {
            isRefreshing = false // 重置刷新状态
          })
      }
      // 刷新状态下，把请求挂起放到 requests 数组中
      return new Promise(resolve => {
        requests.push(() => {
          resolve(request(error.config))
        })
      })
    } else if (status === 403) {
      Message.error('没有权限，请联系管理员')
    } else if (status === 404) {
      Message.error('请求资源不存在')
    } else if (status >= 500) {
      Message.error('服务端错误，请联系管理员')
    }
  } else if (error.request) {
    // 请求发出去了没有收到响应
    Message.error('请求超时，请刷新重试')
  } else {
    // 在设置请求时发生了一些事情，触发了一个错误
    Message.error(`请求失败：${error.message}`)
  }
  // 把请求失败的错误对象继续抛出，扔给上一个调用者
  return Promise.reject(error)
})
function redirectLogin () {
  router.push({
    name: 'login',
    query: {
      redirect: router.currentRoute.fullPath
    }
  })
}
function refreshToken () {
  return axios.create()({
    method: 'POST',
    url: '/front/user/refresh_token',
    data: qs.stringify({
      refreshtoken: store.state.user.refresh_token
    })
  })
}
export default request
// # sourceMappingURL=request.js.map
